Job System
多线程编程计算,主要为了提升性能。
性能分析
Playmode 启动分析 勾选Live实时显示线程信息
只保留脚本
Unity项目依赖的DOTS包安装(基于unity2020 高版本可能直接搜索就能安装)
启动Edit > Project Settings > Package Manager中的Enable Preview Packages
要安装的包没有 可以通过url 下载
最好翻墙
安装com.unity.entities
安装com.unity.burst
安装com.unity.mathematics
安装com.unity.collections
或者直接安装com.unity.rendering.hybrid(依赖上面4个包)
再安装一下com.unity.jobs
Job System语法注意
结构体只包含blittable类型那么结构体创建的变量 也是blittable类型
IJob基础使用
using UnityEngine;
using Unity.Jobs;
using Unity.Mathematics;
using Unity.Collections;
using Unity.Burst;
[BurstCompile]
public struct Job1 : IJob
{
public NativeArray<int> array1;
public NativeArray<int> array2;
public int index;
public void Execute()
{
array2[index] = array1[index] + 1;
Debug.Log(message: $"当前线程 {System.Threading.Thread.CurrentThread.ManagedThreadId}");//打印当前线程ID
}
}
public class MyIJob : MonoBehaviour
{
public int[] testArray = new int[] { 1, 2, 3 };
void Start()
{
Debug.Log(message: $"主线程 {System.Threading.Thread.CurrentThread.ManagedThreadId}");//打印当前线程ID
NativeArray<int> array1 = new NativeArray<int>(testArray, Allocator.TempJob);//非托管区申请内存
NativeArray<int> array2 = new NativeArray<int>(testArray, Allocator.TempJob);//非托管区申请内存
Job1 jb = new Job1() { array1 = array1, array2 = array2, index = 2 };//创建Job
var jobHandle = jb.Schedule();//申请一个线程 计算
jobHandle.Complete();//计算完成
array2.CopyTo(testArray);//计算结构copy到测试数组
array1.Dispose();//释放内存
array2.Dispose();//释放内存
}
}
Allocator.Temp 保存一阵
Allocator.TempJob 保存4帧
Allocator.Persistent 运行期间一直存在
* NativeArray 主动申请内存 使用后需要释放 使用blittable类型 不要用引用类型
只读的可以在多个job中同时读取,可以开启多个job
如果连续开启两个job,想要在job1基础上进行job2(先执行job1后执行job2)可以
IJobParallelForTransform
主要是更新物体的Transform
以下为生成10000个 GameObject 并且朝着一个物体移动,高性能移动物体
using Unity.Collections;
using UnityEngine;
using UnityEngine.Jobs;
//**********************************
//自定义模板创建人:Far
//**********************************
public struct EnemyMove : IJobParallelForTransform
{
public NativeArray<Vector3> Velocity;
public void Execute(int index, TransformAccess transform)
{
transform.position += Velocity[index] * 0.02f;
}
}
using Unity.Collections;
using Unity.Entities;
using Unity.Jobs;
using UnityEngine;
using UnityEngine.Jobs;
//**********************************
//自定义模板创建人:Far
//**********************************
//挂载到一个空物体上
public class SetPos : MonoBehaviour
{
public GameObject enemy;
private TransformAccessArray tranAccessArray;
private const int instanceCount = 10000;
void Start()
{
tranAccessArray = new TransformAccessArray(instanceCount);
for (int i = 0; i < instanceCount; i++)
{
var go = Instantiate(enemy);
go.gameObject.SetActive(true);
go.transform.position = Random.insideUnitSphere * 100;
tranAccessArray.Add(go.transform);
}
}
void Update()
{
var tempVelocity = new NativeArray<Vector3>(instanceCount, Allocator.TempJob);
for (int i = 0; i < tempVelocity.Length; i++)
{
//tempVelocity[i] = Random.insideUnitSphere;
tempVelocity[i] = transform.position - tranAccessArray[i].transform.position;
}
var job = new EnemyMove()
{ Velocity = tempVelocity };
job.Schedule(tranAccessArray)// 这里传入TransformAccessArray
.Complete();
tempVelocity.Dispose();
}
private void OnDestroy()
{
tranAccessArray.Dispose();// 需要Dispose!
}
}